<template>
  <div class="container">
    <slot></slot>
    <div>
      <div class="left-board">
        <el-scrollbar class="left-scrollbar">
          <div class="components-list">
            <div v-for="(item, listIndex) in leftComponents" :key="listIndex">
              <div class="components-title">
                <svg-icon icon-class="component" />
                <span>{{ item.title }}</span>
              </div>
              <template v-if="item.children">
                <el-collapse v-model="activeCollapseNames">
                  <el-collapse-item :title="subItem.title" v-for="(subItem, subIndex) in item.children" :key="subIndex" :name="subItem.title">
                    <component-group :item="subItem" :clone="cloneComponent" @click="addComponent" @end="onEnd" :groupName="item.groupName" :move="item.checkMove"/>
                  </el-collapse-item>
                </el-collapse>
              </template>
              <component-group v-else :item="item" :clone="cloneComponent" @click="addComponent" @end="onEnd" :groupName="item.groupName" :move="item.checkMove"/>
            </div>
          </div>
        </el-scrollbar>
      </div>
      <div class="center-board">
        <!-- <div class="action-bar">
          <el-button icon="el-icon-video-play" type="text" @click="run">
            运行
          </el-button>
          <el-button icon="el-icon-view" type="text" @click="showJson">
            查看json
          </el-button>
          <el-button icon="el-icon-download" type="text" @click="download">
            导出vue文件
          </el-button>
          <el-button class="copy-btn-main" icon="el-icon-document-copy" type="text" @click="copy">
            复制代码
          </el-button>
          <el-button class="delete-btn" icon="el-icon-delete" type="text" @click="empty">
            清空
          </el-button>
        </div> -->
        <el-scrollbar class="center-scrollbar">
          <el-row class="center-board-row" :gutter="formConf.gutter">
            <el-form
              :size="formConf.size"
              :label-position="formConf.labelPosition"
              :disabled="formConf.disabled"
              :label-width="formConf.labelWidth + 'px'"
            >
              <draggable class="drawing-board" :list="drawingList" :animation="340" group="componentsGroup" filter='.undraggable' :move="checkMove">
                <transition-group name="flip-list" tag="div">
                  <draggable-item
                    v-for="(item, index) in drawingList"
                    :key="item.__config__.renderKey"
                    :drawing-list="drawingList"
                    :current-item="item"
                    :index="index"
                    :active-id="activeId"
                    :form-conf="formConf"
                    @activeItem="activeFormItem"
                    @copyItem="drawingItemCopy"
                    @deleteItem="drawingItemDelete"
                    @moveUpItem="drawingItemMoveUp"
                    @moveDownItem="drawingItemMoveDown"
                  />

                </transition-group>
              </draggable>
              <div v-show="drawingList.length === 1 && drawingList[0].__config__.children.cardBody.length === drawingDefalutChildrenLength" class="empty-info">
                从左侧拖入或点选组件进行表单设计
              </div>
            </el-form>
          </el-row>
        </el-scrollbar>
      </div>
      <right-panel
        :active-data="activeData"
        :form-conf="formConf"
        :show-field="!!drawingList.length"
        @tag-change="tagChange"
        @fetch-data="fetchData"
      />
    </div>
    <form-drawer
      :visible.sync="drawerVisible"
      :form-data="formData"
      size="100%"
      :generate-conf="generateConf"
    />
    <json-drawer
      size="60%"
      :visible.sync="jsonDrawerVisible"
      :json-str="JSON.stringify(formData)"
      @refresh="refreshJson"
    />
    <code-type-dialog
      :visible.sync="dialogVisible"
      title="选择生成类型"
      :show-file-name="showFileName"
      @confirm="generate"
    />
    <input id="copyNode" type="hidden">
  </div>
</template>

<script>
import draggable from 'vuedraggable'
import { debounce } from 'throttle-debounce'
import { saveAs } from 'file-saver'
import ClipboardJS from 'clipboard'
import render from '../../components/render/render'
import FormDrawer from './FormDrawer'
import JsonDrawer from './JsonDrawer'
import RightPanel from './RightPanel'
import ComponentGroup from './ComponentGroup.vue'
import {
  workOrderContentComponents, baseComponents, customizedSelectComponents, formConf, customizedSpecialComponents
} from '../../components/generator/config/index.js'
import {
  exportDefault, beautifierConf, isNumberStr, titleCase, deepClone, isObjectObject
} from '../../utils/index'
import {
  makeUpHtml, vueTemplate, vueScript, cssStyle
} from '../../components/generator/html'
import { makeUpJs } from '../../components/generator/js'
import { makeUpCss } from '../../components/generator/css'
import drawingDefalut from '../../components/generator/drawingDefalut'
import logo from '../../assets/logo.png'
import CodeTypeDialog from './CodeTypeDialog'
import DraggableItem from './DraggableItem'
import {
  getDrawingList, saveDrawingList, getIdGlobal, saveIdGlobal, getFormConf
} from '../../utils/db'
import loadBeautifier from '../../utils/loadBeautifier'

let beautifier
const emptyActiveData = { style: {}, autosize: {} }
let oldActiveId
let tempActiveData
const drawingListInDB = getDrawingList()
const formConfInDB = getFormConf()
const idGlobal = getIdGlobal()

export default {
  components: {
    draggable,
    render,
    FormDrawer,
    JsonDrawer,
    RightPanel,
    CodeTypeDialog,
    DraggableItem,
    ComponentGroup
  },
  data() {
    let that = this
    workOrderContentComponents.forEach(t=>{
      t.__config__.group = "workOrderContentComponents"
    })
    function checkCardMove(evt){
      // 获取拖动元素的索引和目标位置的索引
      const dragIndex = evt.draggedContext.index;
      const targetIndex = evt.relatedContext.index;
      // console.log(dragIndex, targetIndex);
      if (targetIndex === 0) {
          return 1;
      }
    }
    function checkItemMove(evt){
      // console.log('evt', evt)
      const targetIndex = evt.relatedContext.index;
      let card = that.findCard(evt.relatedContext.element)
      // console.log('card', card)
      if(card){
        let startDragIndex = card.__config__.startDragIndex
        // console.log('targetIndex',targetIndex, 'startDragIndex', startDragIndex)
        if(startDragIndex !== undefined && targetIndex < startDragIndex-1){
          return false
        }
        if (targetIndex === startDragIndex-1) {
          return 1
        }
      }
    }
    let leftComponents = [
        {
          title: '工单内容配置',
          groupName: 'componentsGroup',
          checkMove: checkCardMove,
          list: workOrderContentComponents
        },
        {
          title: '基础组件',
          list: baseComponents,
          checkMove: checkItemMove
        },
        // 定制组件
        {
          title: '定制组件',
          children: [
            {
              title: '下拉列表',
              list: customizedSelectComponents,
              checkMove: checkItemMove
            },
            {
              title: '特殊组件',
              list: customizedSpecialComponents,
              checkMove: checkItemMove
            }
          ]
        }
    ].map( t=> {
        if(!t.groupName) t.groupName = 'workOrderContentComponents'
        if(!t.checkMove) t.checkMove = function(){}
        return t
    })
    let drawingList = deepClone(drawingDefalut)
    return {
      logo,
      idGlobal,
      formConf,
      workOrderContentComponents,
      baseComponents,
      customizedSelectComponents,
      labelWidth: 100,
      drawingDefalutChildrenLength: drawingList[0].__config__.children.cardBody.length,
      drawingList: drawingList,
      drawingData: {},
      activeId: drawingList[0].formId,
      drawerVisible: false,
      formData: {},
      dialogVisible: false,
      jsonDrawerVisible: false,
      generateConf: null,
      showFileName: false,
      activeData: drawingList[0],
      saveDrawingListDebounce: debounce(340, saveDrawingList),
      saveIdGlobalDebounce: debounce(340, saveIdGlobal),
      leftComponents,
      activeCollapseNames: leftComponents.find(item => item.children)?.children?.map(item => item.title) || [],
    }
  },
  computed: {
  },
  watch: {
    // eslint-disable-next-line func-names
    'activeData.__config__.label': function (val, oldVal) {
      if (
        this.activeData.placeholder === undefined
          || !this.activeData.__config__.tag
          || oldActiveId !== this.activeId
      ) {
        return
      }
      // this.activeData.placeholder = this.activeData.placeholder.replace(oldVal, '') + val
    },
    activeId: {
      handler(val) {
        oldActiveId = val
      },
      immediate: true
    },
    drawingList: {
      handler(val, olVal) {
        //TODO: 等会把下面语句放开
        this.saveDrawingListDebounce(val)
        if (val.length === 0) this.idGlobal = 100
      },
      deep: true
    },
    idGlobal: {
      handler(val) {
        this.saveIdGlobalDebounce(val)
      },
      immediate: true
    }
  },
  created() {
  },
  mounted() {
    if (Array.isArray(drawingListInDB) && drawingListInDB.length > 0) {
      this.drawingList = drawingListInDB
    } else {
      this.drawingList = deepClone(drawingDefalut)
    }
    this.activeFormItem(this.drawingList[0])
    if (formConfInDB) {
      this.formConf = formConfInDB
    }
    loadBeautifier(btf => {
      beautifier = btf
    })
    const clipboard = new ClipboardJS('#copyNode', {
      text: trigger => {
        const codeStr = this.generateCode()
        this.$notify({
          title: '成功',
          message: '代码已复制到剪切板，可粘贴。',
          type: 'success'
        })
        return codeStr
      }
    })
    clipboard.on('error', e => {
      this.$message.error('代码复制失败')
    })
  },
  methods: {
    setObjectValueReduce(obj, strKeys, data) {
      const arr = strKeys.split('.')
      arr.reduce((pre, item, i) => {
        if (arr.length === i + 1) {
          pre[item] = data
        } else if (!isObjectObject(pre[item])) {
          pre[item] = {}
        }
        return pre[item]
      }, obj)
    },
    setRespData(component, resp) {
      const { dataPath, renderKey, dataConsumer } = component.__config__
      if (!dataPath || !dataConsumer) return
      const respData = dataPath.split('.').reduce((pre, item) => pre[item], resp)

      // 将请求回来的数据，赋值到指定属性。
      // 以el-tabel为例，根据Element文档，应该将数据赋值给el-tabel的data属性，所以dataConsumer的值应为'data';
      // 此时赋值代码可写成 component[dataConsumer] = respData；
      // 但为支持更深层级的赋值（如：dataConsumer的值为'options.data'）,使用setObjectValueReduce
      this.setObjectValueReduce(component, dataConsumer, respData)
      const i = this.drawingList.findIndex(item => item.__config__.renderKey === renderKey)
      if (i > -1) this.$set(this.drawingList, i, component)
    },
    fetchData(component) {
      const { dataType, method, url } = component.__config__
      if (dataType === 'dynamic' && method && url) {
        this.setLoading(component, true)
        this.$axios({
          method,
          url
        }).then(resp => {
          this.setLoading(component, false)
          this.setRespData(component, resp.data)
        })
      }
    },
    setLoading(component, val) {
      const { directives } = component
      if (Array.isArray(directives)) {
        const t = directives.find(d => d.name === 'loading')
        if (t) t.value = val
      }
    },
    activeFormItem(currentItem) {
      // console.log('currentItem',currentItem)
      // console.log('drawingList',this.drawingList)
      this.activeData = currentItem
      this.activeId = currentItem.__config__.formId
    },
    findCard(item){
      // console.log('this.drawingList',this.drawingList)
      // console.log('item',item)
      const parentComponent = this.drawingList.find(component => {
        const cardBody = component.__config__.children.cardBody
        return cardBody.includes(item)
      })
      return parentComponent
    },
    //根据当前激活的组件，找到对应的card组件
    activeCard(){
      if(this.activeData?.__config__.tag === 'el-card') return this.activeData
      return this.findCard(this.activeData)
    },
    onEnd(obj) {
      if (obj.from !== obj.to) {
        this.fetchData(tempActiveData)
        this.activeData = tempActiveData
        this.activeId = this.idGlobal
      }
    },
    addComponent(item) {
      const clone = this.cloneComponent(item)
      this.fetchData(clone)
      if(clone.__config__.tag === 'el-card'){
        // 添加在最外层
        this.drawingList.push(clone)
      }else{
        // 添加到激活的卡片中
        this.activeCard()?.__config__.children.cardBody.push(clone)
      }
      this.activeFormItem(clone)
    },
    cloneComponent(origin) {
      const clone = deepClone(origin)
      const config = clone.__config__
      config.span = this.formConf.span // 生成代码时，会根据span做精简判断
      this.createIdAndKey(clone)
      // clone.placeholder !== undefined && (clone.placeholder += config.label)
      tempActiveData = clone
      return tempActiveData
    },
    createIdAndKey(item) {
      const config = item.__config__
      config.formId = ++this.idGlobal
      config.renderKey = `${config.formId}${+new Date()}` // 改变renderKey后可以实现强制更新组件
      if (config.layout === 'colFormItem' || config.layout === 'tsElTabs') {
        item.__vModel__ = `field${this.idGlobal}`
      } else if (config.layout === 'rowFormItem') {
        config.componentName = `row${this.idGlobal}`
        !Array.isArray(config.children) && (config.children = [])
        delete config.label // rowFormItem无需配置label属性
      }
      if (Array.isArray(config.children)) {
        config.children = config.children.map(childItem => this.createIdAndKey(childItem))
      }
      return item
    },
    AssembleFormData() {
      this.formData = {
        fields: deepClone(this.drawingList),
        ...this.formConf
      }
    },
    generate(data) {
      const func = this[`exec${titleCase(this.operationType)}`]
      this.generateConf = data
      func && func(data)
    },
    execRun(data) {
      this.AssembleFormData()
      this.drawerVisible = true
    },
    execDownload(data) {
      const codeStr = this.generateCode()
      const blob = new Blob([codeStr], { type: 'text/plain;charset=utf-8' })
      saveAs(blob, data.fileName)
    },
    execCopy(data) {
      document.getElementById('copyNode').click()
    },
    empty() {
      this.$confirm('确定要清空所有组件吗？', '提示', { type: 'warning' }).then(
        () => {
          // this.drawingList = []
          // this.drawingList.splice(1)
          this.drawingList = deepClone(drawingDefalut)
          console.log('this.drawingList',this.drawingList)
          this.idGlobal = 100
        }
      )
    },
    drawingItemCopy(item, list) {
      let clone = deepClone(item)
      clone = this.createIdAndKey(clone)
      list.push(clone)
      this.activeFormItem(clone)
    },
    drawingItemDelete(index, list) {
      list.splice(index, 1)
      this.$nextTick(() => {
        const len = this.drawingList.length
        if (len) {
          this.activeFormItem(this.drawingList[len - 1])
        }
      })
    },
    drawingItemMoveUp(index, list) {
      if (index === 0) return
      const prev = list[index - 1]
      this.$set(list, index - 1, list[index])
      this.$set(list, index, prev)
    },
    drawingItemMoveDown(index, list) {
      if (index === list.length - 1) return
      const next = list[index + 1]
      this.$set(list, index + 1, list[index])
      this.$set(list, index, next)
    },
    generateCode() {
      const { type } = this.generateConf
      this.AssembleFormData()
      const script = vueScript(makeUpJs(this.formData, type))
      const html = vueTemplate(makeUpHtml(this.formData, type))
      const css = cssStyle(makeUpCss(this.formData))
      return beautifier.html(html + script + css, beautifierConf.html)
    },
    showJson() {
      this.AssembleFormData()
      this.jsonDrawerVisible = true
    },
    download() {
      this.dialogVisible = true
      this.showFileName = true
      this.operationType = 'download'
    },
    run() {
      this.dialogVisible = true
      this.showFileName = false
      this.operationType = 'run'
    },
    copy() {
      this.dialogVisible = true
      this.showFileName = false
      this.operationType = 'copy'
    },
    tagChange(newTag) {
      newTag = this.cloneComponent(newTag)
      const config = newTag.__config__
      newTag.__vModel__ = this.activeData.__vModel__
      config.formId = this.activeId
      config.span = this.activeData.__config__.span
      this.activeData.__config__.tag = config.tag
      this.activeData.__config__.tagIcon = config.tagIcon
      this.activeData.__config__.document = config.document
      if (typeof this.activeData.__config__.defaultValue === typeof config.defaultValue) {
        config.defaultValue = this.activeData.__config__.defaultValue
      }
      Object.keys(newTag).forEach(key => {
        if (this.activeData[key] !== undefined) {
          newTag[key] = this.activeData[key]
        }
      })
      this.activeData = newTag
      this.updateDrawingList(newTag, this.drawingList)
    },
    updateDrawingList(newTag, list) {
      const index = list.findIndex(item => item.__config__.formId === this.activeId)
      if (index > -1) {
        list.splice(index, 1, newTag)
      } else {
        list.forEach(item => {
          if (Array.isArray(item.__config__.children)) this.updateDrawingList(newTag, item.__config__.children)
        })
      }
    },
    refreshJson(data) {
      this.drawingList = deepClone(data.fields)
      delete data.fields
      this.formConf = data
    },
    checkMove(evt) {
      // 获取拖动元素的索引和目标位置的索引
      const dragIndex = evt.draggedContext.index;
      const targetIndex = evt.relatedContext.index;

      // 如果目标位置的索引为 0，则阻止拖动元素的移动
      if (targetIndex === 0) {
          return false;
      }
    }
  }
}
</script>

<style lang='scss'>
@import '../../styles/home';
</style>
<style lang='scss' scoped>
::v-deep .el-collapse-item__header{
  font-size: 12px;
  height: 30px;
  line-height: 30px;
  padding-left: 17px;
}
::v-deep .el-collapse-item__content {
  padding-bottom: 0;
}
::v-deep .box-card .el-card__body:after {
      content: "";
      display: block;
      clear: both;
}
</style>
